{-# LANGUAGE ScopedTypeVariables #-} module Travel (stepCreature, turnCreature, ClimbOutcome, resolveClimb, executeClimb, TeleportJumpOutcome, resolveTeleportJump, executeTeleportJump) where import Control.Monad.Maybe import Facing import DB import Plane import Data.Maybe import Control.Monad import Control.Monad.Trans import Data.Ord import Position import TerrainData import Data.List (minimumBy) import Creature import CreatureData import Logging import TravelData walkCreature :: (DBReadable db) => Facing -> (Integer,Integer) -> Location CreatureRef () -> db (Location CreatureRef ()) walkCreature face (x',y') l = liftM (fromMaybe l) $ runMaybeT $ do (plane_ref,Position (x,y)) <- MaybeT $ return $ extractParent l let standing = Standing { standing_plane = plane_ref, standing_position = Position (x+x',y+y'), standing_facing = face } is_passable <- lift $ isTerrainPassable plane_ref (child l) (standing_position standing) when (not is_passable) $ do lift $ logDB log_travel WARNING $ "Terrain not passable." fail "" return $ generalizeParent $ toStanding standing l stepCreature :: (DBReadable db) => Facing -> Location CreatureRef () -> db (Location CreatureRef ()) stepCreature face = walkCreature face (facingToRelative face) turnCreature :: (DBReadable db) => Facing -> Location CreatureRef () -> db (Location CreatureRef ()) turnCreature face = walkCreature face (0,0) -------------------------------------------------------------------------------- -- Travel between planes. -------------------------------------------------------------------------------- data ClimbOutcome = ClimbGood ClimbDirection CreatureRef Standing | ClimbFailed -- | -- Climb up or down between Planes. -- resolveClimb :: (DBReadable db) => CreatureRef -> ClimbDirection -> db ClimbOutcome resolveClimb creature_ref direction = liftM (fromMaybe ClimbFailed) $ runMaybeT $ do l <- lift $ dbWhere creature_ref ((p,pos) :: (PlaneRef,Position)) <- MaybeT $ return $ extractParent l terrain_type <- lift $ terrainAt p pos let (expected_starting_terrain, expected_landing_terrain) = case direction of ClimbUp -> (Upstairs,Downstairs) ClimbDown -> (Downstairs,Upstairs) when (terrain_type /= expected_starting_terrain) $ do lift $ logDB log_travel WARNING $ "Not standing on correct stairway." fail "" lift $ logDB log_travel DEBUG $ "Stepping " ++ show direction ++ " from: " ++ show (p,pos) let face = fromMaybe Here $ extractParent l p' <- MaybeT $ case direction of ClimbDown -> getBeneath p ClimbUp -> liftM extractParent $ dbWhere p lift $ logDB log_travel DEBUG $ "Stepping " ++ show direction ++ " to: " ++ show p' pos' <- lift $ pickRandomClearSite 10 0 0 pos (== expected_landing_terrain) p' return $ ClimbGood direction creature_ref $ Standing { standing_plane = p', standing_position = pos', standing_facing = face } -- | Execute a resolved climb attempt. executeClimb :: ClimbOutcome -> DB () executeClimb ClimbFailed = return () executeClimb (ClimbGood direction creature_ref standing_location) = do _ <- dbMove (return . toStanding standing_location) creature_ref dbPushSnapshot $ ClimbEvent direction creature_ref return () -------------------------------------------------------------------------------- -- Teleportation/Jumping -------------------------------------------------------------------------------- -- | -- Try to teleport the creature to the specified Position. The teleport attempt can be automatically retried a number of times, and the most accurate attempt will be used. -- If the retries are negative, the teleport will be made artificially innacurate. -- randomTeleportLanding :: (DBReadable db) => Integer -> PlaneRef -> Position -> Position -> db Position randomTeleportLanding retries plane_ref source_destination goal_destination = do landings <- replicateM (fromInteger $ max 1 retries) $ (pickRandomClearSite 3) 0 0 goal_destination (not . (`elem` impassable_terrains)) plane_ref return $ minimumBy (comparing $ \p -> distanceBetweenSquared goal_destination p ^ 2 * distanceBetweenSquared source_destination p) landings data TeleportJumpOutcome = TeleportJumpGood CreatureRef Standing | TeleportJumpFailed -- | -- Teleport jump a creature about 7 units in the specified direction. -- resolveTeleportJump :: (DBReadable db) => CreatureRef -> Facing -> db TeleportJumpOutcome resolveTeleportJump creature_ref face = liftM (fromMaybe TeleportJumpFailed) $ runMaybeT $ do start_location <- lift $ dbWhere creature_ref jump_roll <- liftM roll_log $ lift $ rollCreatureAbilityScore JumpSkill 0 (child start_location) standing_location <- MaybeT $ return $ extractParent start_location landing_position <- lift $ randomTeleportLanding jump_roll (standing_plane standing_location) (standing_position standing_location) $ offsetPosition (facingToRelative7 face) $ standing_position standing_location case () of () | jump_roll <= 0 -> return TeleportJumpFailed () | otherwise -> return $ TeleportJumpGood (child start_location) $ standing_location { standing_position = landing_position, standing_facing = face } -- | Execute a resolved teleport jump. executeTeleportJump :: TeleportJumpOutcome -> DB () executeTeleportJump TeleportJumpFailed = return () executeTeleportJump (TeleportJumpGood creature_ref standing_location) = do _ <- dbMove (return . toStanding standing_location) creature_ref dbPushSnapshot $ TeleportEvent creature_ref return ()