Welcome to the Builder Academy

Question Immunities, resistances and vulnerabilities

More
05 Mar 2025 19:29 #10572 by Banlock
One thing I like about tabletop D&D is the idea of immunities, resistances and vulnerabilities. I'm sure you folks are familiar, but I'm talking about monsters being invulnerable to non-magical weapons, or taking half damage from piercing weapons, or being extra vulnerable to cold. This seemed to add an extra strategic level to combat. 

I don't see anything in tbaMUD that handles this and I'd like to add it. Has anyone out there already done this? I'd hate to re-invent the wheel if there is already a tested solution. I understand this will make my world files incompatible (as I will have to store the information somewhere), but I'm willing to take that step. A starting point might be "attack_hit_type" maybe?

By the way, I asked ChatGTP about tbaMUD and it seemed somewhat familiar with the project :-)

In loose outline, here is what it proposed:

1. Modify the char_data Structure (In structs.h, expand the char_data structure to store resistances and vulnerabilities)

2. Define Damage Types (In constants.c, define new damage types (if they don’t exist already))

3. Modify the Damage Calculation (In fight.c, locate damage() function and modify it to check resistances and vulnerabilities)

4. Add New Mob Attributes (Modify db.c to allow setting resistances/vulnerabilities in the mob files (e.g., via the mob editor))

5. Update Mob Loading Code (Modify db.c to read these new attributes when loading mobs from files)

6. Modify Object Weapon Effects (If you want weapons to deal specific damage types, modify weapon.c or the obj_data struct to track dam_type)

7. Test and Debug (Implement logging in fight.c to ensure damage calculations reflect the new mechanics)

If anyone wants to see the full conversation - including sample code - I have a link here:  chatgpt.com/share/67c89a8c-0224-800d-ab74-64542a4763b2 . The conversation also includes me asking ChatCGT what it knows about tabMUD.

What do you folks think? :-)

Please Log in or Create an account to join the conversation.

More
06 Mar 2025 19:08 - 10 Mar 2025 02:42 #10573 by Salty
I did this.  Below is most of the code.


The defines in structs.h:
Code:
#define RESIST_UNARMED 0 #define RESIST_EXOTIC 1 #define RESIST_BLUNT 2 #define RESIST_PIERCE 3 #define RESIST_SLASH 4 #define RESIST_MAGIC 5 #define NUM_RESISTS 6 #define APPLY_RESIST_UNARMED 32 #define APPLY_RESIST_EXOTIC 33 #define APPLY_RESIST_BLUNT 34 #define APPLY_RESIST_PIERCE 35 #define APPLY_RESIST_SLASH 36 #define APPLY_RESIST_MAGIC 37

In char_point_data in structs.h
Code:
  int resists[NUM_RESISTS];

In utils.h
Code:
#define GET_RESISTS(ch, i) ((ch)->points.resists[i])

In apply_types[] in const.c
Code:
  "RESIST_UNARMED",   "RESIST_EXOTIC",   "RESIST_BLUNT",   "RESIST_PIERCE",   "RESIST_SLASH",   "RESIST_MAGIC",

In damage() in fight.c:
Code:
  /* attacktype is passed to the damage() function by hit() or mag_damage() */   if (attacktype >= TYPE_HIT && attacktype < TYPE_HIT + NUM_ATTACK_TYPES)   {     switch (attacktype)     {     case TYPE_HIT:     case TYPE_THRASH:     case TYPE_PUNCH:       resist = RESIST_UNARMED;       break;     case TYPE_CRUSH:     case TYPE_SHIELD_SLAM:     case TYPE_BLAST:       resist = RESIST_EXOTIC;       break;     case TYPE_BLUDGEON:     case TYPE_POUND:     case TYPE_MAUL:       resist = RESIST_BLUNT;       break;     case TYPE_STING:     case TYPE_BITE:     case TYPE_PIERCE:     case TYPE_STAB:       resist = RESIST_PIERCE;       break;     case TYPE_WHIP:     case TYPE_SLASH:     case TYPE_CLAW:       resist = RESIST_SLASH;       break;     default:       resist = -1;       break;     }   }   /* All damaging spells go through this function and process them here */   if (attacktype >= 0 && attacktype < MAX_SPELLS)   {     if (spell_info[attacktype].violent)       resist = RESIST_MAGIC;     if (ROOM_FLAGGED(IN_ROOM(ch), ROOM_MAGIC_DAMPEN))       dampen = TRUE;     else if (ROOM_FLAGGED(IN_ROOM(victim), ROOM_MAGIC_DAMPEN))       dampen = TRUE;   }   /* We grab the leftover attacks here */   if (resist > -1 && dam >= 2)   {     for (int r = 0; r < NUM_RESISTS; r++)     {       if (resist == r)       {         if (GET_RESISTS(victim, r) > 0)           dam -= dam * MIN(GET_RESISTS(victim, r), 100) / 100;         if (GET_RESISTS(victim, r) < 0)           dam += dam * MIN(abs(GET_RESISTS(victim, r)), 100) / 100;       }     }   }

In medit_parse() in medit.c
Code:
  case MEDIT_RESIST_0:     GET_RESISTS(OLC_MOB(d), RESIST_UNARMED) = LIMIT(i, -100, 100);     write_to_output(d, "Enter the melee resist exotic (-100 to 100):  ");     OLC_MODE(d) = MEDIT_RESIST_1;     return;   case MEDIT_RESIST_1:     GET_RESISTS(OLC_MOB(d), RESIST_EXOTIC) = LIMIT(i, -100, 100);     write_to_output(d, "Enter the melee resist blunt (-100 to 100):  ");     OLC_MODE(d) = MEDIT_RESIST_2;     return;   case MEDIT_RESIST_2:     GET_RESISTS(OLC_MOB(d), RESIST_BLUNT) = LIMIT(i, -100, 100);     write_to_output(d, "Enter the melee resist pierce (-100 to 100):  ");     OLC_MODE(d) = MEDIT_RESIST_3;     return;   case MEDIT_RESIST_3:     GET_RESISTS(OLC_MOB(d), RESIST_PIERCE) = LIMIT(i, -100, 100);     write_to_output(d, "Enter the melee resist slash (-100 to 100):  ");     OLC_MODE(d) = MEDIT_RESIST_4;     return;  case MEDIT_RESIST_4:     GET_RESISTS(OLC_MOB(d), RESIST_SLASH) = LIMIT(i, -100, 100);     write_to_output(d, "Enter the magic resist (-100 to 100):  ");     OLC_MODE(d) = MEDIT_RESIST_5;     return;   case MEDIT_RESIST_5:     GET_RESISTS(OLC_MOB(d), RESIST_MAGIC) = LIMIT(i, -100, 100);     OLC_VAL(d) = TRUE;     medit_disp_stats_menu(d);     return;

In interpret_espec() in db.c
Code:
CASE("RES_0")   {     RANGE(-100, 100);     mob_proto[i].points.resists[RESIST_UNARMED] = num_arg;   }   CASE("RES_1")   {     RANGE(-100, 100);     mob_proto[i].points.resists[RESIST_EXOTIC] = num_arg;   }   CASE("RES_2")   {     RANGE(-100, 100);     mob_proto[i].points.resists[RESIST_BLUNT] = num_arg;   }   CASE("RES_3")   {     RANGE(-100, 100);     mob_proto[i].points.resists[RESIST_PIERCE] = num_arg;   }   CASE("RES_4")   {     RANGE(-100, 100);     mob_proto[i].points.resists[RESIST_SLASH] = num_arg;   }   CASE("RES_5")   {     RANGE(-100, 100);     mob_proto[i].points.resists[RESIST_MAGIC] = num_arg;   } 

In write_mobile_espec() in genmob.c
Code:
  if (GET_RESISTS(mob, 0) != 0)     fprintf(fd, "RES_0: %d\n", GET_RESISTS(mob,RESIST_UNARMED));   if (GET_RESISTS(mob, 1) != 0)     fprintf(fd, "RES_1: %d\n", GET_RESISTS(mob,RESIST_EXOTIC));   if (GET_RESISTS(mob, 2) != 0)     fprintf(fd, "RES_2: %d\n", GET_RESISTS(mob,RESIST_BLUNT));   if (GET_RESISTS(mob, 3) != 0)     fprintf(fd, "RES_3: %d\n", GET_RESISTS(mob,RESIST_PIERCE));   if (GET_RESISTS(mob, 4) != 0)     fprintf(fd, "RES_4: %d\n", GET_RESISTS(mob,RESIST_SLASH));   if (GET_RESISTS(mob, 5) != 0)     fprintf(fd, "RES_5: %d\n", GET_RESISTS(mob,RESIST_MAGIC));

In aff_apply_modify() in handler.c
Code:
  case APPLY_RESIST_UNARMED:     GET_RESISTS(ch, RESIST_UNARMED) += mod;     break;   case APPLY_RESIST_EXOTIC:     GET_RESISTS(ch, RESIST_EXOTIC) += mod;     break;   case APPLY_RESIST_BLUNT:     GET_RESISTS(ch, RESIST_BLUNT) += mod;     break;   case APPLY_RESIST_PIERCE:     GET_RESISTS(ch, RESIST_PIERCE) += mod;     break;   case APPLY_RESIST_SLASH:     GET_RESISTS(ch, RESIST_SLASH) += mod;     break;     case APPLY_RESIST_MAGIC:     GET_RESISTS(ch, RESIST_MAGIC) += mod;     break;

In load_char() in players.c:
Code:
        else if (!strcmp(tag, "RES0"))           GET_RESISTS(ch, RESIST_UNARMED) = atoi(line);         else if (!strcmp(tag, "RES1"))           GET_RESISTS(ch, RESIST_EXOTIC) = atoi(line);         else if (!strcmp(tag, "RES2"))           GET_RESISTS(ch, RESIST_BLUNT) = atoi(line);         else if (!strcmp(tag, "RES3"))           GET_RESISTS(ch, RESIST_PIERCE) = atoi(line);         else if (!strcmp(tag, "RES4"))           GET_RESISTS(ch, RESIST_SLASH) = atoi(line);         else if (!strcmp(tag, "RES5"))           GET_RESISTS(ch, RESIST_MAGIC) = atoi(line);

I added them to medit_disp_stats_menu() as well but my display has been entirely reorganized so providing the code won't be so helpful.

I added them to do_score() in act.informative.c.  It's been reorganized as well.

I added them to zaffs[], set_fields[], perform_set() and do_stat_character() in act.wizard.c.  This is fairly straightforward.

There are some spells in magic.c (stoneskin, steelskin, etc) than can buff the resists.

These changes shouldn't break your mob or player files as interpret_espec() and load_char() only process the value if the value exists.  Once you load the player or mob and modify the values, the appropriate file will be updated and saved.  

Additionally, because I added the APPLY_RESIST_(TYPE), the resistances can be added to equipment with oedit.  A wooden shield could make you resistant to pierce but vulnerable to blunt.  A dragonscale shield could make you resistant to exotic and magic.

All in all, this has worked flawlessly and the only change I'm pondering is reorganizing my damage spells into magical colors like MtG or classical greek elements and adding the corresponding resistances.

Note:  negative values for resists are calculated as vulnerabilities.

Hope this helps.
Last edit: 10 Mar 2025 02:42 by Salty. Reason: Formatting, comments and additional information
The following user(s) said Thank You: ironfist

Please Log in or Create an account to join the conversation.

More
06 Mar 2025 19:38 - 06 Mar 2025 19:39 #10574 by ironfist
You could also breakdown spells further if you needed to (mine already has a few extra fields like cooldown, spell_level for memorization, etc):
Code:
static void spello(int spl, const char *name, int max_mana, int min_mana,     int mana_change, int minpos, int spell_level, int cooldown, int targets, int violent,     int routines, int damage_type, const char *wearoff); #define skillo(skill, name, cooldown, damage_type) spello(skill, name, 0, 0, 0, 0, 0, \                                           cooldown, 0, 0, 0, damage_type, NULL);   spello(SPELL_BLACK_LIGHTNING, "black lightning", 75, 55, 5, POS_FIGHTING,   7, 3, TAR_CHAR_ROOM | TAR_FIGHT_VICT, TRUE, MAG_DAMAGE,   DAM_DEATH | DAM_ELECTRICITY,         NULL); structs.h:646:#define DAM_NONE            (1 << 0) structs.h:647:#define DAM_FIRE            (1 << 1) /* burning hands, fireball */ structs.h:648:#define DAM_COLD            (1 << 2) /* cone of cold, ice */ structs.h:649:#define DAM_ELECTRICITY    (1 << 3) /* lightning */ structs.h:650:#define DAM_ACID            (1 << 4) /* acid arrow, black dragon breath */ structs.h:651:#define DAM_POISON          (1 << 5) /* sting/bite/poison */ structs.h:652:#define DAM_FORCE          (1 << 6) /* magic missile */ structs.h:653:#define DAM_PSYCHIC        (1 << 7) /* psionic blast */ structs.h:654:#define DAM_DEATH          (1 << 8) /* necrotic/undead/energy drain */ structs.h:655:#define DAM_SONIC          (1 << 9) /* thunder/shatter */ structs.h:656:#define DAM_HOLY            (1 << 10) /* radiant/smite */ structs.h:657:#define DAM_PIERCE          (1 << 11) /* spear/bite/dagger */ structs.h:658:#define DAM_SLASH          (1 << 12) /* sword/talon */ structs.h:659:#define DAM_BLUDGEONING    (1 << 13) /* mace/hammer/club */
Last edit: 06 Mar 2025 19:39 by ironfist. Reason: add code block
The following user(s) said Thank You: Salty

Please Log in or Create an account to join the conversation.

More
06 Mar 2025 19:43 #10576 by Salty
Yeah I'm going to go down a very similar path with the magic resists and add a specific resist for each dam type.

Please Log in or Create an account to join the conversation.

More
12 Mar 2025 23:37 #10583 by zusuk
There is a codebase built on top of TBA that has this along with a lot of pathfinder/d20 stuff thrown in:

github.com/LuminariMUD/Luminari-Source

Website
www.luminariMUD.com

Main Game Port
luminariMUD.com:4100
The following user(s) said Thank You: Banlock, ironfist

Please Log in or Create an account to join the conversation.

Time to create page: 0.216 seconds