module Game.Werewolf.Role (
Role,
name, allegiance, balance, description, rules,
Allegiance(..),
_Angel, _Villagers, _Werewolves,
allRoles, restrictedRoles,
allAllegiances,
devotedServantRole, wildChildRole, wolfHoundRole,
angelRole,
defenderRole, druidRole, jesterRole, scapegoatRole, seerRole, simpleVillagerRole,
villagerVillagerRole, witchRole,
simpleWerewolfRole,
is, isn't, filteredBy,
) where
import Control.Lens hiding (isn't)
import Data.Function
import Data.List
import Data.Monoid
import Data.Text (Text)
import qualified Data.Text as T
data Role = Role
{ _name :: Text
, _allegiance :: Allegiance
, _balance :: Int
, _description :: Text
, _rules :: Text
} deriving (Read, Show)
data Allegiance = Angel | Villagers | Werewolves
deriving (Eq, Read, Show)
makeLenses ''Role
instance Eq Role where
(==) = (==) `on` view name
makePrisms ''Allegiance
allRoles :: [Role]
allRoles =
[ angelRole
, defenderRole
, devotedServantRole
, druidRole
, jesterRole
, scapegoatRole
, seerRole
, simpleVillagerRole
, simpleWerewolfRole
, villagerVillagerRole
, wildChildRole
, witchRole
, wolfHoundRole
]
restrictedRoles :: [Role]
restrictedRoles = allRoles \\ [simpleVillagerRole, simpleWerewolfRole]
allAllegiances :: [Allegiance]
allAllegiances = [Angel, Villagers, Werewolves]
devotedServantRole :: Role
devotedServantRole = Role
{ _name = "Devoted Servant"
, _allegiance = Villagers
, _balance = 2
, _description = T.unwords
[ "Who could dream of a better servant than one willing to give up her life for that of her"
, "masters? Don't rejoice too fast, as the devouring ambition within her could spell the"
, "end of the village!"
]
, _rules = T.unwords
[ "Before the revelation of the card of the player eliminated by the village's vote, the"
, "Devoted Servant can reveal herself by showing her card and taking on the role of the"
, "eliminated player. Upon taking on her master's role, any special abilities are reset and"
, "any first turns are replayed."
]
}
wildChildRole :: Role
wildChildRole = Role
{ _name = "Wild-child"
, _allegiance = Villagers
, _balance = 1
, _description = T.unwords
[ "Abandoned in the woods by his parents at a young age, he was raised by wolves. As soon"
, "as he learned how to walk on all fours, the Wild-child began to wander around Miller's"
, "Hollow. One day, fascinated by an inhabitant of the village who was walking upright"
, "with grace and presence, he made them his secret role model. He then decided to"
, "integrate himself into the community of Miller's Hollow and entered, worried, in the"
, "village. The community was moved by his frailty, adopted him, and welcomed him in their"
, "fold. What will become of him: honest Villager or terrible Werewolf? For all of his"
, "life, the heart of the Wild-child will swing between these two alternatives. May his"
, "model confirm him in his newfound humanity."
]
, _rules = T.unwords
[ "On the first night, the Wild-child may choose a player to become his role model. If"
, "during the game the chosen player is eliminated, the Wild-child becomes a Werewolf. He"
, "will then wake up the next night with his peers and will devour with them each night"
, "until the end of the game. However for as long as the Wild-child's role model is alive,"
, "he remains a Villager."
]
}
wolfHoundRole :: Role
wolfHoundRole = Role
{ _name = "Wolf-hound"
, _allegiance = Villagers
, _balance = 1
, _description = T.unwords
[ "All dogs know in the depths of their soul that their ancestors were wolves and that it's"
, "mankind who has kept them in the state of childishness and fear, the faithful and"
, "generous companions. In any case, only the Wolf-hound can decide if he'll obey his human"
, "and civilized master or if he'll listen to the call of wild nature buried within him."
]
, _rules = T.unwords
[ "On the first night, the Wolf-hound chooses if he wants to be a Simple Villager or"
, "Werewolf. The choice is final."
]
}
angelRole :: Role
angelRole = Role
{ _name = "Angel"
, _allegiance = Angel
, _balance = 0
, _description = T.unwords
[ "The muddy life of a village infested with evil creatures repulses him; he wishes to"
, "believe he's the victim of a terrible nightmare, in order to finally wake up in his"
, "comfortable bed."
]
, _rules = T.intercalate "\n"
[ T.unwords
[ "When the Angel is in play, the game always begins with the village's debate followed"
, "by an elimination vote, and then the first night."
]
, T.unwords
[ "The Angel wins if he manages to get eliminated on the first round (day or night). If"
, "he fails, then he becomes a Simple Villager for the rest of the game."
]
]
}
defenderRole :: Role
defenderRole = Role
{ _name = "Defender"
, _allegiance = Villagers
, _balance = 2
, _description =
"This character can save the Villagers from the bite of the Werewolves."
, _rules = T.intercalate "\n"
[ T.unwords
[ "Each night the Defender is called before the Werewolves to select a player deserving"
, "of his protection. That player is safe during the night (and only that night)"
, "against the Werewolves."
]
, "The Defender may not protect the same person two nights in a row."
]
}
druidRole :: Role
druidRole = Role
{ _name = "Druid"
, _allegiance = Villagers
, _balance = 2
, _description = T.unwords
[ "How honoured we are to be in the presence of such a noble leader. The return of the"
, "Druid marks an exceptional time in Fougères's history! Friend of the woodland creatures,"
, "practiced philosopher and now, with the help of Ferina their companion, a bane to the"
, "Werewolves themselves! My does she have a nose on her, strong enough to sniff out"
, "lycanthropes in close proximity! Listen for her grunt and heed the Druid's words for"
, "they will not let you down."
]
, _rules = T.unwords
[ "Each morning when Ferina wakes from her slumber she will be alert and cautious. If the"
, "Druid is next to a Werewolf then Ferina will grunt in warning."
]
}
scapegoatRole :: Role
scapegoatRole = Role
{ _name = "Scapegoat"
, _allegiance = Villagers
, _balance = 0
, _description = T.unwords
[ "Werewolves don't just spring up out of the ground! That's where dwarves come from."
, "Clearly someone is to blame for this affliction to Fougères. Ah! Unluckily, since no-one"
, "actually knows who brought them here, the blame is always laid upon the Scapegoat."
]
, _rules = T.intercalate "\n"
[ T.unwords
[ "If the village's vote ends in a tie, it's the Scapegoat who is eliminated instead of"
, "no-one."
]
, T.unwords
[ "In this event, the Scapegoat has one last task to complete: he must choose whom is"
, "permitted to vote or not on the next day."
]
]
}
seerRole :: Role
seerRole = Role
{ _name = "Seer"
, _allegiance = Villagers
, _balance = 2
, _description = T.unwords
[ "Frequently misunderstood and thought to be a fortune teller, the Seer has the ability to"
, "see into fellow townsfolk and determine their true nature. This ability to see is not"
, "given out lightly, for certain it is a gift! Visions will always be true, but only for"
, "the present as not even the Seer knowns what the future holds."
]
, _rules = "Each night the Seer sees the allegiance of a player of their choice."
}
simpleVillagerRole :: Role
simpleVillagerRole = Role
{ _name = "Simple Villager"
, _allegiance = Villagers
, _balance = 1
, _description = T.unwords
[ "A simple, ordinary townsperson in every way. Some may be cobblers, others bakers or even"
, "nobles. No matter their differences though, the plight of Werewolves in Fougères unites"
, "them in this unfortunate time."
]
, _rules = T.unwords
[ "A Simple Villager has no special abilities, they must use their guile to determine whom"
, "among them is not who they say they are."
]
}
jesterRole :: Role
jesterRole = Role
{ _name = "Jester"
, _allegiance = Villagers
, _balance = 0
, _description = T.unwords
[ "Every village needs a Jester, they're so stupid but provide so much entertainment! The"
, "Jester may not have any special abilities, but at least no one in the village would want"
, "to hurt him."
]
, _rules = T.intercalate "\n"
[ T.unwords
[ "If the village votes to lynch the Jester, his identity is revealed. The village"
, "realise there's no point in burning him and so he is set free."
]
, T.unwords
[ "The Jester continues to play but may no longer vote as no one can take him"
, "seriously."
]
]
}
villagerVillagerRole :: Role
villagerVillagerRole = Role
{ _name = "Villager-Villager"
, _allegiance = Villagers
, _balance = 2
, _description = T.unwords
[ "This person has a soul as clear and transparent as the water from a mountain stream."
, "They will deserve the attentive ear of their peers and will make their word decisive in"
, "crucial moments."
]
, _rules = T.unwords
[ "When the game begins, the village is told the identity of the Villager-Villager, thus"
, "ensuring certainty that its owner is truly an innocent Villager."
]
}
witchRole :: Role
witchRole = Role
{ _name = "Witch"
, _allegiance = Villagers
, _balance = 3
, _description = T.unwords
[ "Somehow forgotten with the coming of the Werewolves, the Witch has a chance to prove"
, "themself valuable to the village. The Witch is blessed (or maybe cursed) with the"
, "ability to make two powerful potions; one of which may heal a victim of the Werewolves,"
, "the other able to poison a player. The use of these potions could alter the village's"
, "currently misguided perception of the Witch."
]
, _rules = T.unwords
[ "The Witch is called after the Werewolves. They are able to heal and poison one player"
, "per game. There is no restriction on using both potions in one night or to heal"
, "themself."
]
}
simpleWerewolfRole :: Role
simpleWerewolfRole = Role
{ _name = "Simple Werewolf"
, _allegiance = Werewolves
, _balance = 4
, _description = T.unwords
[ "It's difficult to tell, but a Werewolf is merely a Villager underneath; unfortunately"
, "one that has been afflicted with lycanthropy and consequently transforms each night into"
, "a fearsome lupine. Not all hope is yet lost, some Villagers would like to try and cure"
, "the curse of lycanthropy. Perhaps one day this dream will be realised."
]
, _rules = "A Werewolf may never devour another Werewolf."
}
is :: Getting Any s a -> s -> Bool
is query = has query
isn't :: Getting All s a -> s -> Bool
isn't query = hasn't query
filteredBy :: Eq b => Lens' a b -> b -> Traversal' a a
filteredBy lens value = filtered ((value ==) . view lens)