feat(xanthous): Add configurable natural attacks
Allow configuring the natural attacks (eg, part of their body rather than an item) of a creature. Each attack has a description and a damage associated with it. Change-Id: I69698a8ac4ee2da91e4c88e419593627519522a5 Reviewed-on: https://cl.tvl.fyi/c/depot/+/3220 Reviewed-by: grfn <grfn@gws.fyi> Tested-by: BuildkiteCI
This commit is contained in:
parent
88b0d0eecc
commit
fb5bec8d95
7 changed files with 69 additions and 12 deletions
|
@ -24,7 +24,10 @@ import Xanthous.Entities.Creature.Hippocampus
|
||||||
import Xanthous.Entities.Character (Character)
|
import Xanthous.Entities.Character (Character)
|
||||||
import qualified Xanthous.Entities.Character as Character
|
import qualified Xanthous.Entities.Character as Character
|
||||||
import qualified Xanthous.Entities.RawTypes as Raw
|
import qualified Xanthous.Entities.RawTypes as Raw
|
||||||
import Xanthous.Entities.RawTypes (CreatureType, HasLanguage (language), getLanguage)
|
import Xanthous.Entities.RawTypes
|
||||||
|
( CreatureType, HasLanguage(language), getLanguage
|
||||||
|
, HasAttacks (attacks)
|
||||||
|
)
|
||||||
import Xanthous.Game.State
|
import Xanthous.Game.State
|
||||||
import Xanthous.Game.Lenses
|
import Xanthous.Game.Lenses
|
||||||
( entitiesCollision, collisionAt
|
( entitiesCollision, collisionAt
|
||||||
|
@ -36,6 +39,7 @@ import Xanthous.Random
|
||||||
import Xanthous.Monad (say)
|
import Xanthous.Monad (say)
|
||||||
import Xanthous.Generators.Speech (word)
|
import Xanthous.Generators.Speech (word)
|
||||||
import qualified Linear.Metric as Metric
|
import qualified Linear.Metric as Metric
|
||||||
|
import qualified Xanthous.Messages as Messages
|
||||||
--------------------------------------------------------------------------------
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
-- TODO move the following two classes to a more central location
|
-- TODO move the following two classes to a more central location
|
||||||
|
@ -86,7 +90,7 @@ stepGormlak ticks pe@(Positioned pos creature) = do
|
||||||
$ creature ^. field @"_hippocampus" . destination
|
$ creature ^. field @"_hippocampus" . destination
|
||||||
let progress' =
|
let progress' =
|
||||||
dest ^. destinationProgress
|
dest ^. destinationProgress
|
||||||
+ creature ^. field @"_creatureType" . Raw.speed . invertedRate |*| ticks
|
+ creatureType ^. Raw.speed . invertedRate |*| ticks
|
||||||
if progress' < 1
|
if progress' < 1
|
||||||
then pure
|
then pure
|
||||||
$ pe'
|
$ pe'
|
||||||
|
@ -106,10 +110,17 @@ stepGormlak ticks pe@(Positioned pos creature) = do
|
||||||
when (any (entityIs @Character) ents) attackCharacter
|
when (any (entityIs @Character) ents) attackCharacter
|
||||||
pure pe'
|
pure pe'
|
||||||
where
|
where
|
||||||
|
creatureType = creature ^. field @"_creatureType"
|
||||||
vision = visionRadius creature
|
vision = visionRadius creature
|
||||||
attackCharacter = do
|
attackCharacter = do
|
||||||
say ["combat", "creatureAttack"] $ object [ "creature" A..= creature ]
|
attack <- choose $ creatureType ^. attacks
|
||||||
character %= Character.damage 1
|
attackDescription <- Messages.render (attack ^. Raw.description)
|
||||||
|
$ object []
|
||||||
|
say ["combat", "creatureAttack"]
|
||||||
|
$ object [ "creature" A..= creature
|
||||||
|
, "attackDescription" A..= attackDescription
|
||||||
|
]
|
||||||
|
character %= Character.damage (attack ^. Raw.damage)
|
||||||
|
|
||||||
yellAtCharacter = for_ (creature ^. field @"_creatureType" . language)
|
yellAtCharacter = for_ (creature ^. field @"_creatureType" . language)
|
||||||
$ \lang -> do
|
$ \lang -> do
|
||||||
|
|
|
@ -13,6 +13,8 @@ module Xanthous.Entities.RawTypes
|
||||||
-- ** Language
|
-- ** Language
|
||||||
, LanguageName(..)
|
, LanguageName(..)
|
||||||
, getLanguage
|
, getLanguage
|
||||||
|
-- ** Attacks
|
||||||
|
, Attack(..)
|
||||||
|
|
||||||
-- * Items
|
-- * Items
|
||||||
, ItemType(..)
|
, ItemType(..)
|
||||||
|
@ -25,6 +27,7 @@ module Xanthous.Entities.RawTypes
|
||||||
, isWieldable
|
, isWieldable
|
||||||
|
|
||||||
-- * Lens classes
|
-- * Lens classes
|
||||||
|
, HasAttacks(..)
|
||||||
, HasAttackMessage(..)
|
, HasAttackMessage(..)
|
||||||
, HasChar(..)
|
, HasChar(..)
|
||||||
, HasDamage(..)
|
, HasDamage(..)
|
||||||
|
@ -52,6 +55,7 @@ import Xanthous.Data (TicksPerTile, Hitpoints)
|
||||||
import Xanthous.Data.EntityChar
|
import Xanthous.Data.EntityChar
|
||||||
import Xanthous.Util.QuickCheck
|
import Xanthous.Util.QuickCheck
|
||||||
import Xanthous.Generators.Speech (Language, gormlak, english)
|
import Xanthous.Generators.Speech (Language, gormlak, english)
|
||||||
|
import Xanthous.Orphans ()
|
||||||
--------------------------------------------------------------------------------
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
-- | Identifiers for languages that creatures can speak.
|
-- | Identifiers for languages that creatures can speak.
|
||||||
|
@ -73,6 +77,23 @@ getLanguage :: LanguageName -> Language
|
||||||
getLanguage Gormlak = gormlak
|
getLanguage Gormlak = gormlak
|
||||||
getLanguage English = english
|
getLanguage English = english
|
||||||
|
|
||||||
|
-- | Natural attacks for creature types
|
||||||
|
data Attack = Attack
|
||||||
|
{ -- | the @{{creature}}@ @{{description}}@
|
||||||
|
_description :: !Message
|
||||||
|
-- | Damage dealt
|
||||||
|
, _damage :: !Hitpoints
|
||||||
|
}
|
||||||
|
deriving stock (Show, Eq, Ord, Generic)
|
||||||
|
deriving anyclass (NFData, CoArbitrary, Function)
|
||||||
|
deriving Arbitrary via GenericArbitrary Attack
|
||||||
|
deriving (ToJSON, FromJSON)
|
||||||
|
via WithOptions '[ FieldLabelModifier '[Drop 1]
|
||||||
|
, OmitNothingFields 'True
|
||||||
|
]
|
||||||
|
Attack
|
||||||
|
makeFieldsNoPrefix ''Attack
|
||||||
|
|
||||||
data CreatureType = CreatureType
|
data CreatureType = CreatureType
|
||||||
{ _name :: !Text
|
{ _name :: !Text
|
||||||
, _description :: !Text
|
, _description :: !Text
|
||||||
|
@ -81,8 +102,10 @@ data CreatureType = CreatureType
|
||||||
, _friendly :: !Bool
|
, _friendly :: !Bool
|
||||||
, _speed :: !TicksPerTile
|
, _speed :: !TicksPerTile
|
||||||
, _language :: !(Maybe LanguageName)
|
, _language :: !(Maybe LanguageName)
|
||||||
, _sayVerb :: !(Maybe Text) -- ^ The verb, in present tense, for when the
|
, -- | The verb, in present tense, for when the creature says something
|
||||||
-- creature says something
|
_sayVerb :: !(Maybe Text)
|
||||||
|
, -- | The creature's natural attacks
|
||||||
|
_attacks :: !(NonNull (Vector Attack))
|
||||||
}
|
}
|
||||||
deriving stock (Show, Eq, Ord, Generic)
|
deriving stock (Show, Eq, Ord, Generic)
|
||||||
deriving anyclass (NFData, CoArbitrary, Function)
|
deriving anyclass (NFData, CoArbitrary, Function)
|
||||||
|
|
|
@ -2,8 +2,8 @@ Creature:
|
||||||
name: gormlak
|
name: gormlak
|
||||||
description: a gormlak
|
description: a gormlak
|
||||||
longDescription: |
|
longDescription: |
|
||||||
A chittering imp-like creature with bright yellow horns. It adores shiny objects
|
A chittering imp-like creature with bright yellow horns and sharp claws. It
|
||||||
and gathers in swarms.
|
adores shiny objects and gathers in swarms.
|
||||||
char:
|
char:
|
||||||
char: g
|
char: g
|
||||||
style:
|
style:
|
||||||
|
@ -13,3 +13,8 @@ Creature:
|
||||||
friendly: false
|
friendly: false
|
||||||
language: Gormlak
|
language: Gormlak
|
||||||
sayVerb: yells
|
sayVerb: yells
|
||||||
|
attacks:
|
||||||
|
- description:
|
||||||
|
- claws you
|
||||||
|
- slashes you with its claws
|
||||||
|
damage: 1
|
||||||
|
|
|
@ -10,3 +10,6 @@ Creature:
|
||||||
maxHitpoints: 3
|
maxHitpoints: 3
|
||||||
speed: 100
|
speed: 100
|
||||||
friendly: false
|
friendly: false
|
||||||
|
attacks:
|
||||||
|
- description: slams into you
|
||||||
|
damage: 1
|
||||||
|
|
|
@ -9,6 +9,7 @@ module Xanthous.Messages
|
||||||
-- * Game messages
|
-- * Game messages
|
||||||
, messages
|
, messages
|
||||||
, render
|
, render
|
||||||
|
, render_
|
||||||
, lookup
|
, lookup
|
||||||
, message
|
, message
|
||||||
, message_
|
, message_
|
||||||
|
@ -17,7 +18,7 @@ module Xanthous.Messages
|
||||||
import Xanthous.Prelude hiding (lookup)
|
import Xanthous.Prelude hiding (lookup)
|
||||||
--------------------------------------------------------------------------------
|
--------------------------------------------------------------------------------
|
||||||
import Control.Monad.Random.Class (MonadRandom)
|
import Control.Monad.Random.Class (MonadRandom)
|
||||||
import Data.Aeson (FromJSON, ToJSON, toJSON)
|
import Data.Aeson (FromJSON, ToJSON, toJSON, object)
|
||||||
import qualified Data.Aeson as JSON
|
import qualified Data.Aeson as JSON
|
||||||
import Data.Aeson.Generic.DerivingVia
|
import Data.Aeson.Generic.DerivingVia
|
||||||
import Data.FileEmbed
|
import Data.FileEmbed
|
||||||
|
@ -89,6 +90,10 @@ render msg params = do
|
||||||
tpl <- resolve msg
|
tpl <- resolve msg
|
||||||
pure . toStrict . renderMustache tpl $ toJSON params
|
pure . toStrict . renderMustache tpl $ toJSON params
|
||||||
|
|
||||||
|
-- | Render a message with an empty set of params
|
||||||
|
render_ :: (MonadRandom m) => Message -> m Text
|
||||||
|
render_ msg = render msg $ object []
|
||||||
|
|
||||||
lookup :: [Text] -> Message
|
lookup :: [Text] -> Message
|
||||||
lookup path = fromMaybe notFound $ messages ^? ix path
|
lookup path = fromMaybe notFound $ messages ^? ix path
|
||||||
where notFound
|
where notFound
|
||||||
|
|
|
@ -300,9 +300,21 @@ deriving stock instance Ord Attr
|
||||||
|
|
||||||
--------------------------------------------------------------------------------
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
instance (SemiSequence a, Arbitrary (Element a), Arbitrary a)
|
||||||
|
=> Arbitrary (NonNull a) where
|
||||||
|
arbitrary = ncons <$> arbitrary <*> arbitrary
|
||||||
|
|
||||||
|
instance ToJSON a => ToJSON (NonNull a) where
|
||||||
|
toJSON = toJSON . toNullable
|
||||||
|
|
||||||
|
instance (FromJSON a, MonoFoldable a) => FromJSON (NonNull a) where
|
||||||
|
parseJSON = maybe (fail "Found empty list") pure . fromNullable <=< parseJSON
|
||||||
|
|
||||||
instance NFData a => NFData (NonNull a) where
|
instance NFData a => NFData (NonNull a) where
|
||||||
rnf xs = xs `seq` toNullable xs `deepseq` ()
|
rnf xs = xs `seq` toNullable xs `deepseq` ()
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
instance forall t name. (NFData t, Monoid t, NFData name)
|
instance forall t name. (NFData t, Monoid t, NFData name)
|
||||||
=> NFData (Editor t name) where
|
=> NFData (Editor t name) where
|
||||||
rnf ed = getName @_ @name ed `deepseq` getEditContents ed `deepseq` ()
|
rnf ed = getName @_ @name ed `deepseq` getEditContents ed `deepseq` ()
|
||||||
|
|
|
@ -71,9 +71,7 @@ combat:
|
||||||
generic:
|
generic:
|
||||||
- You hit the {{creature.creatureType.name}}.
|
- You hit the {{creature.creatureType.name}}.
|
||||||
- You attack the {{creature.creatureType.name}}.
|
- You attack the {{creature.creatureType.name}}.
|
||||||
creatureAttack:
|
creatureAttack: The {{creature.creatureType.name}} {{attackDescription}}
|
||||||
- The {{creature.creatureType.name}} hits you!
|
|
||||||
- The {{creature.creatureType.name}} attacks you!
|
|
||||||
killed:
|
killed:
|
||||||
- You kill the {{creature.creatureType.name}}!
|
- You kill the {{creature.creatureType.name}}!
|
||||||
- You've killed the {{creature.creatureType.name}}!
|
- You've killed the {{creature.creatureType.name}}!
|
||||||
|
|
Loading…
Reference in a new issue