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:
Griffin Smith 2021-06-18 12:42:42 -04:00 committed by grfn
parent 88b0d0eecc
commit fb5bec8d95
7 changed files with 69 additions and 12 deletions

View file

@ -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

View file

@ -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)

View file

@ -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

View file

@ -10,3 +10,6 @@ Creature:
maxHitpoints: 3 maxHitpoints: 3
speed: 100 speed: 100
friendly: false friendly: false
attacks:
- description: slams into you
damage: 1

View file

@ -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

View file

@ -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` ()

View file

@ -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}}!