]> granicus.if.org Git - nethack/commitdiff
fix B17005 - autoquiver bug
authornethack.rankin <nethack.rankin>
Sat, 25 Jan 2003 04:46:11 +0000 (04:46 +0000)
committernethack.rankin <nethack.rankin>
Sat, 25 Jan 2003 04:46:11 +0000 (04:46 +0000)
     hold_another_object() will try to quiver the object being held if
it's a weapon (or gem/rock ammo) and the autoquiver option is enabled and
the quiver is empty.  It was doing that even if the object had just merged
with primary or secondary weapon, resulting in it being equipped in two
slots at once.  (Easiest way to reproduce it is via wish+wield+wish for
similar item, but it could also occur when stealing while in nymph form.)

     This also addresses one of the old items on <Someone>'s list:  if you
carry a sling equipped in the alternate weapon slot, include gems and
rocks as likely candidates for quivering same as when wielding a sling.
This extends that to autoquivering; ammo appropriate for your alternate
is now given preference over arbitrary weapons (ammo for your wielded
weapon and arbitrary missiles still take precedence over alternate ammo).

     Bug?  pickup_object() is not autoquiver aware, hence autopickup isn't
either.

     Bug too (perhaps moot if the above is changed)?  Snagging a monster's
weapon with a bullwhip uses hold_another_object() so possibly autoquivers;
snagging an object off the floor with a grappling hook uses pickup_object()
so doesn't.

doc/fixes34.1
src/dothrow.c
src/invent.c
src/wield.c

index 4e3eb75032194a0df40553306b46f479fc38435a..d580cec63d516f8266852dc5afeafcce797e9294 100644 (file)
@@ -362,6 +362,9 @@ engraving on headstone will appropriately dull your weapon
 certain types of golems should not "catch fire" so adjust the messages
 no longer need to manually examine inventory after regaining sight in order
        to give a type name to an object picked up while blind
+when adding an object to inventory, it is possible for it to becomed both
+       wielded and quivered if it merges with weapon and autoquiver is enabled
+include rocks as likely candidates for quivering if alternate weapon is a sling
 
 
 Platform- and/or Interface-Specific Fixes
index 8520ffb95f24b75bb33bfe01e441b0636e0cf153..b1ec8230e4ee3712ab1b2da236c23e7dd7ada4ff 100644 (file)
@@ -1,4 +1,4 @@
-/*     SCCS Id: @(#)dothrow.c  3.4     2003/01/08      */
+/*     SCCS Id: @(#)dothrow.c  3.4     2003/01/24      */
 /* Copyright (c) Stichting Mathematisch Centrum, Amsterdam, 1985. */
 /* NetHack may be freely redistributed.  See license for details. */
 
@@ -215,7 +215,7 @@ dothrow()
 static void
 autoquiver()
 {
-       register struct obj *otmp, *oammo = 0, *omissile = 0, *omisc = 0;
+       struct obj *otmp, *oammo = 0, *omissile = 0, *omisc = 0, *altammo = 0;
 
        if (uquiver)
            return;
@@ -233,6 +233,8 @@ autoquiver()
                         objects[otmp->otyp].oc_material == GLASS)) {
                if (uslinging())
                    oammo = otmp;
+               else if (ammo_and_launcher(otmp, uswapwep))
+                   altammo = otmp;
                else if (!omisc)
                    omisc = otmp;
            } else if (otmp->oclass == GEM_CLASS) {
@@ -242,6 +244,8 @@ autoquiver()
                if (ammo_and_launcher(otmp, uwep))
                    /* Ammo matched with launcher (bow and arrow, crossbow and bolt) */
                    oammo = otmp;
+               else if (ammo_and_launcher(otmp, uswapwep))
+                   altammo = otmp;
                else
                    /* Mismatched ammo (no better than an ordinary weapon) */
                    omisc = otmp;
@@ -263,6 +267,8 @@ autoquiver()
            setuqwep(oammo);
        else if (omissile)
            setuqwep(omissile);
+       else if (altammo)
+           setuqwep(altammo);
        else if (omisc)
            setuqwep(omisc);
 
index e1e965b0ef5c0c8a8118a6d428fc16b76c4a5c73..2ec4d433c0b8bd918794d7bcba2d4b4fd290c79d 100644 (file)
@@ -1,4 +1,4 @@
-/*     SCCS Id: @(#)invent.c   3.4     2002/12/13      */
+/*     SCCS Id: @(#)invent.c   3.4     2003/01/24      */
 /* Copyright (c) Stichting Mathematisch Centrum, Amsterdam, 1985. */
 /* NetHack may be freely redistributed.  See license for details. */
 
@@ -405,39 +405,35 @@ const char *drop_fmt, *drop_arg, *hold_msg;
            }
        }
        if (Fumbling) {
-               if (drop_fmt) pline(drop_fmt, drop_arg);
-               dropy(obj);
+           if (drop_fmt) pline(drop_fmt, drop_arg);
+           dropy(obj);
        } else {
-               long oquan = obj->quan;
-               int prev_encumbr = near_capacity();     /* before addinv() */
-
-               /* encumbrance only matters if it would now become worse
-                  than max( current_value, stressed ) */
-               if (prev_encumbr < MOD_ENCUMBER) prev_encumbr = MOD_ENCUMBER;
-               if (drop_arg) {
-                       /* addinv() may redraw the entire inventory, overwriting
-                        * drop_arg when it comes from something like doname()
-                        */
-                       Strcpy(buf, drop_arg);
-                       drop_arg = buf;
-               }
-               obj = addinv(obj);
-               if (inv_cnt() > 52
+           long oquan = obj->quan;
+           int prev_encumbr = near_capacity(); /* before addinv() */
+
+           /* encumbrance only matters if it would now become worse
+              than max( current_value, stressed ) */
+           if (prev_encumbr < MOD_ENCUMBER) prev_encumbr = MOD_ENCUMBER;
+           /* addinv() may redraw the entire inventory, overwriting
+              drop_arg when it comes from something like doname() */
+           if (drop_arg) drop_arg = strcpy(buf, drop_arg);
+
+           obj = addinv(obj);
+           if (inv_cnt() > 52
                    || ((obj->otyp != LOADSTONE || !obj->cursed)
                        && near_capacity() > prev_encumbr)) {
-                       if (drop_fmt) pline(drop_fmt, drop_arg);
-                       /* undo any merge which took place */
-                       if (obj->quan > oquan) {
-                           obj = splitobj(obj, oquan);
-                       }
-                       dropx(obj);
-               } else {
-                       if (flags.autoquiver && !uquiver &&
-                           (is_missile(obj) ||
-                            (uwep && ammo_and_launcher(obj, uwep))))
-                           setuqwep(obj);
-                       if (hold_msg || drop_fmt) prinv(hold_msg, obj, oquan);
-               }
+               if (drop_fmt) pline(drop_fmt, drop_arg);
+               /* undo any merge which took place */
+               if (obj->quan > oquan) obj = splitobj(obj, oquan);
+               dropx(obj);
+           } else {
+               if (flags.autoquiver && !uquiver && !obj->owornmask &&
+                       (is_missile(obj) ||
+                           ammo_and_launcher(obj, uwep) ||
+                           ammo_and_launcher(obj, uswapwep)))
+                   setuqwep(obj);
+               if (hold_msg || drop_fmt) prinv(hold_msg, obj, oquan);
+           }
        }
        return obj;
 }
index 70ddeaf161409e72d074a16022070dfd65791b59..92e84cf91cf98ae2df5cd120d5c0e57c8bc29a41 100644 (file)
@@ -331,18 +331,20 @@ int
 dowieldquiver()
 {
        register struct obj *newquiver;
-
+       const char *quivee_types = (uslinging() ||
+                 (uswapwep && objects[uswapwep->otyp].oc_skill == P_SLING)) ?
+                                 bullets : ready_objs;
 
        /* Since the quiver isn't in your hands, don't check cantwield(), */
        /* will_weld(), touch_petrifies(), etc. */
        multi = 0;
 
        /* Because 'Q' used to be quit... */
-       if (!flags.suppress_alert || flags.suppress_alert < FEATURE_NOTICE_VER(3,3,0))
+       if (flags.suppress_alert < FEATURE_NOTICE_VER(3,3,0))
                pline("Note: Please use #quit if you wish to exit the game.");
 
        /* Prompt for a new quiver */
-       if (!(newquiver = getobj(uslinging() ? bullets : ready_objs, "ready")))
+       if (!(newquiver = getobj(quivee_types, "ready")))
                /* Cancelled */
                return (0);