{-# LANGUAGE DeriveGeneric #-}
module Game.LambdaHack.Common.Point
( Point(..), PointI
, chessDist, euclidDistSq, adjacent, bla, fromTo
, originPoint
, speedupHackXSize
#ifdef EXPOSE_INTERNAL
, blaXY, balancedWord
#endif
) where
import Prelude ()
import Game.LambdaHack.Core.Prelude
import Data.Binary
import Data.Int (Int32)
import qualified Data.Primitive.PrimArray as PA
import GHC.Generics (Generic)
import Game.LambdaHack.Definition.Defs
speedupHackXSize :: PA.PrimArray X
{-# NOINLINE speedupHackXSize #-}
speedupHackXSize = PA.primArrayFromList [80]
data Point = Point
{ px :: X
, py :: Y
}
deriving (Eq, Ord, Generic)
instance Show Point where
show (Point x y) = show (x, y)
instance Binary Point where
put = put . (fromIntegral :: Int -> Int32) . fromEnum
get = fmap (toEnum . (fromIntegral :: Int32 -> Int)) get
instance Enum Point where
fromEnum Point{..} =
let !xsize = PA.indexPrimArray speedupHackXSize 0
in
#ifdef WITH_EXPENSIVE_ASSERTIONS
assert (px >= 0 && py >= 0 && px < xsize
`blame` "invalid point coordinates"
`swith` (px, py))
#endif
(px + py * xsize)
toEnum n = let !xsize = PA.indexPrimArray speedupHackXSize 0
(py, px) = n `quotRem` xsize
in Point{..}
type PointI = Int
chessDist :: Point -> Point -> Int
chessDist (Point x0 y0) (Point x1 y1) = max (abs (x1 - x0)) (abs (y1 - y0))
euclidDistSq :: Point -> Point -> Int
euclidDistSq (Point x0 y0) (Point x1 y1) =
(x1 - x0) ^ (2 :: Int) + (y1 - y0) ^ (2 :: Int)
adjacent :: Point -> Point -> Bool
{-# INLINE adjacent #-}
adjacent s t = chessDist s t == 1
bla :: X -> Y -> Int -> Point -> Point -> Maybe [Point]
bla rXmax rYmax eps source target =
if source == target then Nothing
else Just $
let inBounds p@(Point x y) =
rXmax > x && x >= 0 && rYmax > y && y >= 0 && p /= source
in takeWhile inBounds $ tail $ blaXY eps source target
blaXY :: Int -> Point -> Point -> [Point]
blaXY eps (Point x0 y0) (Point x1 y1) =
let (dx, dy) = (x1 - x0, y1 - y0)
xyStep b (x, y) = (x + signum dx, y + signum dy * b)
yxStep b (x, y) = (x + signum dx * b, y + signum dy)
(p, q, step) | abs dx > abs dy = (abs dy, abs dx, xyStep)
| otherwise = (abs dx, abs dy, yxStep)
bw = balancedWord p q (eps `mod` max 1 q)
walk w xy = xy : walk (tail w) (step (head w) xy)
in map (uncurry Point) $ walk bw (x0, y0)
balancedWord :: Int -> Int -> Int -> [Int]
balancedWord p q eps | eps + p < q = 0 : balancedWord p q (eps + p)
balancedWord p q eps = 1 : balancedWord p q (eps + p - q)
fromTo :: Point -> Point -> [Point]
fromTo (Point x0 y0) (Point x1 y1) =
let fromTo1 :: Int -> Int -> [Int]
fromTo1 z0 z1
| z0 <= z1 = [z0..z1]
| otherwise = [z0,z0-1..z1]
result
| x0 == x1 = map (Point x0) (fromTo1 y0 y1)
| y0 == y1 = map (`Point` y0) (fromTo1 x0 x1)
| otherwise = error $ "diagonal fromTo"
`showFailure` ((x0, y0), (x1, y1))
in result
originPoint :: Point
originPoint = Point 0 0