forked from crawl/crawl
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmutation_creation.txt
317 lines (268 loc) · 12.9 KB
/
mutation_creation.txt
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
Written by Cerol, 2017/02/09
Before we get to modding species in my set of tutorials, we need to
cover mutations and abilities first. All of a species innate abilities
are handled by mutations. Some grant stat changes, others grant
abilities, but they all use the same framework.
We start off by editing mutation-type.h to add the entry for our new
mutation. This time, we're updating mutation_data.h in the same way,
putting our new entry right above the NUM_MUTATIONS magic entry. Note
that if you're adding multiple mutations (or anything, really) it
doesn't matter which order they get added in, as long as they don't
replace the spot any existing mutation had.
#if TAG_MAJOR_VERSION == 34
MUT_STURDY_FRAME,
MUT_SANGUINE_ARMOUR,
#endif
MUT_LASER_EYES, //first new entry
MUT_PREHENSILE_BEARD, //second new entry
NUM_MUTATIONS,
RANDOM_MUTATION,
There's more special cases for mutations below RANDOM_MUTATION than we
saw for jobs, but those are for code that adds or removes mutations in
play. We aren't concerned with those now.
The second place we'll handle code is mutation-data.h. Much less happens
here than in other structs of similar purpose, but it's similarly nicely
documented. Most of the actual work for mutations is checked in the
relevant part of the code elsewhere in the game. This mostly handles how
many levels your mutation has and the text for them.
struct mutation_def
{
mutation_type mutation;
short weight; ///< Commonality of the mutation;
/// bigger = appears more often.
short levels; ///< The number of levels of the mutation.
mutflags uses; ///< Bitfield holding types of effects that
/// grant this mutation (mutflag::*)
bool form_based; ///< Mutation is suppressed when shapechanged.
const char* short_desc; ///< What appears on the '%' screen.
const char* have[3]; ///< What appears on the 'A' screen.
const char* gain[3]; ///< Message when you gain the mutation.
const char* lose[3]; ///< Message when you lose the mutation.
};
Again, copy and paste an existing one to minimize mistakes in structure:
{ MUT_LASER_EYES, 2, 2, mutflag::GOOD, false,
"laser eyes",
{"You can shoot laser beams from your eyes.",
"You can shoot BIGGER laser beams from your eyes.",
""},
{"Your eyes burn with power.",
"Your burning eyes intensify."
""},
{"Your eyes feel refreshed and cease burning.",
"Your eyes relax and burn less.",
""},
},
{ MUT_PREHENSILE_BEARD, 1, 1, mutflag::GOOD, false,
"prehensile beard",
{"Your beard can move around on its own.",
"",
""},
{"Your beard gains a will of its own.",
""
""},
{"Your beard shaves itself off in shame.",
"",
""},
},
We're going to have the laser eyes mutation be an activateable ability,
and prehensile beard will be yet another off-hand attack passive power.
These aren't fantastically clever mutations, but they'll illustrate how
to implement your own.
We can look at the naga's spit poison ability as the closest thing in
place to our laser eyes power. We'll search for that mutation's name
[MUT_SPIT_POISON] and we see that it's checked in ability.cc,
mutation.cc, and tags.cc outside of the mutation and species definition
blocks. Lucky for us, tsgs.cc is for save file compatibility across
versions, so we can ignore it for our new mutations. Were we editing
levels of an existing mutation, we might want to update tags.cc as well
to handle saves to keep them in line with the current range of levels.
For the laser eyes ability, we need to add an entry in ability-type.h
for an activated ability. This list is nicely organized right now, but
we should add our entry to the end right before NUM_ABILITIES to
minimize any possible save compatibility issue. I'll skip cutting and
pasting yet another block of code, but our name is ABIL_LASER_EYES.
Next, in ability.cc, we find that there's an array called Ability_List[].
The entries in it aren't sorted the same as the enum, but that's ok.
We'll still add a new entry to the bottom. The actual entries follow
this struct:
// Structure for representing an ability:
struct ability_def
{
ability_type ability;
const char * name;
unsigned int mp_cost; // magic cost of ability
scaling_cost hp_cost; // hit point cost of ability
generic_cost piety_cost; // + random2((piety_cost + 1) / 2 + 1)
failure_info failure; // calculator for failure odds
ability_flags flags; // used for additional cost notices
};
Here we see there's 2 different random rolls involving different costs,
2 static costs that won't change, and flags for other costs. Let's add
our own, similar to ABIL_SPIT_POISON:
{ ABIL_LASER_EYES, "Shoot Eye Lasers",
1, 0, 50, 0, {FAIL_XL, 20, 1}, abflag::NONE },
This ability will cost us 1 MP to use, and the success chance is only attached
to our character level, not any skill. Farther down, in, the _do_ability
function, we see the master switch() statement that handles ability use. We'll
add ours in here, and order does not matter for this one, so we'll place it by
similar abilities instead of at the end:
case ABIL_LASER_EYES: // eye lasers
{
int power = 25 + (you.experience_level * 2 * you.get_mutation_level(MUT_LASER_EYES);
beam.range = _calc_breath_ability_range(abil.ability);
if (you.get_mutation_level(MUT_MISSING_EYE) //Ru - Sacrifice an Eye
power = power / 2; //only get half power if you only have half as many eyes.
if (you.get_mutation_level(MUT_EYEBALLS)) // Jiyva mutation.
power = power * 4; //if you're all eyes, you're also all eye-lasers!
if (!spell_direction(abild, beam)
|| !player_tracer(ZAP_DISINTEGRATE, power, beam))
{
return spret::abort;
}
else
{
fail_check();
zapping(ZAP_DISINTEGRATE, power, beam);
}
break;
}
In this case, I'm being lazy and using an existing spell for my laser
eyes' effect, and the casting power gets slightly stronger every level,
and doubles if I have 2 levels instead of 1. I could create a new spell,
and a new zap_type entry to match it, but that's another post.
You may also want to update the your_talents() or find_ability_slot()
blocks of ability.cc to set the letter for your ability if it should
have its own static letter. We won't this time, so laser eyes will
simply go in the next open letter slot for the character. In addition,
we also need to add an entry to _calc_breath_ability_range() for our
ranged ability.
case ABIL_BREATHE_STEAM: return 6;
case ABIL_SPIT_POISON: return 5;
case ABIL_BREATHE_POISON: return 6;
case ABIL_LASER_EYES: return 7; //7 is max for Line of
//Sight, so that's how far
//our eyes can work.
If you have an ability that might not always be possible to activate
(like a breath weapon, or stopping flight over deep water), you need to
go to _check_ability_possible() and add a check there. Laser eyes
shouldn't have any such conditions. We may also need to add the ability
as a talent (yet another term, this one means 'activateable ability' as
I've been using it here) in the your_talents function like so:
if (you.get_mutation_level(MUT_LASER_EYES) > 0)
_add_talent(talents, ABIL_LASER_EYES, check_confused);
You can do all sorts of complicated checks here, and the vampire part
should be a good example of how complex you might want to get at worst.
Note that deity-granted talents will be added automatically, and don't
appear in this block of code. We could also edit find_ability_slot() if
we wanted to assign a set letter to our ability, but I don't currently
want to do that for a randomly acquired mutation.
That should be all of the required steps for an active ability.
Now, let's look at our passive power, prehensile beard. Again, this is a
1-level copy of the horns/hooves/beak/etc mutations for demonstration
purposes, so we can search for MUT_HORNS and see it gets checked in
itemuse.cc, melee_attack.cc, mutation.cc, tiledoll.cc, and transform.cc
in addition to the files we handled at the beginning.
transform.cc handles gaining MUT_HORNS via the Beastly Appendage spell,
so we won't need to copy that part. All of the mutation.cc entries
involve conflicting mutations and gear slots, and the item-use.cc
entries stop you from equipping gear that won't fit over horns. This is
worth taking a look at, especially if you want to do similar body mods
like new scale mutations, or blade-hands, or something else that should
replace a piece of gear. We'll assume that beards can be tucked through
the facemask or bottom of a helmet and won't conflict with equipment.
tiledoll.cc handles the player character's tile icon in tiles mode. We
should update that to be a good citizen, but I don't want to worry about
the complexity of displaying a layered icon correctly in this tutorial.
That leaves melee_attack.cc as where the meat of the mutation goes. The
good news is that we can make a copy of already frequently implemented
code. First let's copy this auxiliary attack class and modify it, then
make an instance of our new class, and then check for our new attack
along with the other auxiliary attacks:
//CREATE THIS:
class AuxBeard: public AuxAttackType
{
public:
AuxBeard()
: AuxAttackType(9, "beard-slap") { }; //base damage, then display name.
int get_damage() const override
{
//dwarven beards are just as strong as their dwarf in many cases.
return damage + you.species == SP_DEEP_DWARF ? 20 : 1;
}
};
//ALSO CREATE THIS
static const AuxBeard AUX_BEARD = AuxBeard();
//UPDATE BOTTOM OF THIS LIST
static const AuxAttackType* const aux_attack_types[] =
{
&AUX_CONSTRICT,
&AUX_KICK,
&AUX_HEADBUTT,
&AUX_PECK,
&AUX_TAILSLAP,
&AUX_PUNCH,
&AUX_BITE,
&AUX_PSEUDOPODS,
&AUX_TENTACLES,
&AUX_BEARD,
};
//ADD A CASE TO THIS FUNCTION:
bool melee_attack::_extra_aux_attack(unarmed_attack_type atk)
{
if (atk != UNAT_CONSTRICT
&& you.strength() + you.dex() <= random2(50))
{
return false;
}
switch (atk)
{
case UNAT_CONSTRICT:
return you.get_mutation_level(MUT_CONSTRICTING_TAIL)
|| you.species == SP_OCTOPODE && you.has_usable_tentacle();
case UNAT_BEARD:
return you.get_mutation_level(MUT_PREHENSILE_BEARD)
//REST OF THE FUNCTION TRIMMED FOR SPACE
Now we have to go back to melee_attack.h and add in the new UNAT_BEARD
enum entry to unarmed_attack_type, since we didn't see that until getting
to the code above:
enum unarmed_attack_type
{
UNAT_NO_ATTACK, // 0
UNAT_CONSTRICT, // put constriction first so octopodes will use it
UNAT_KICK,
UNAT_HEADBUTT,
UNAT_PECK,
UNAT_TAILSLAP,
UNAT_PUNCH,
UNAT_BITE,
UNAT_PSEUDOPODS,
UNAT_TENTACLES,
UNAT_BEARD,
UNAT_FIRST_ATTACK = UNAT_CONSTRICT,
UNAT_LAST_ATTACK = UNAT_BEARD, //we had to update this now because
//we replaced the last attack. Still
//stay before NUM_UNARMED_ATTACKS
NUM_UNARMED_ATTACKS,
};
...and that should do it. Compile, and you should have a small chance
for a character to gain laser eyes or an off-hand attack with a beard
when mutating. Other flags can mark your mutation as being usable
exclusively by a deity.
Your mutations can do pretty much anything, so your own new mutations
might be much more complicated than these examples are. There are enough
current mutations that you most likely will have a good starting place
by looking at an existing mutation and making small changes to it. The
most complexity comes from when a mutation gets referenced on something
else, like how our beard ended up needed a new class to work correctly.
Summary Notes:
- mutation-type.h for mutation_type enum entry
- mutation-data.h to fill in the basics of the mutation
- ability-type.h for ability_type enum entry
- mutation.cc for any conflicting mutations or mutations sharing the
same 'slot'.
- ability.cc for mutations that have an activatable component (talents).
- mutations can affect stuff in almost any other file, depending on
what your mutation changes. Our example changed melee_attack.cc.
- check for your mutation using
"you.get_mutation_level(MUT_NAME_HERE) > 0"
as a simple check when getting creative with your code.