status hilites: Hp and Pw percentages
The hitpoints and power/energy status values (and corresponding maxima) shown on the screen are capped at 9999 to control status line width. The actual values can be bigger than that. Highlights based on percentages were doing their calculations on the potentially truncated values rather than on the actual ones. Add another field to the blstat[] structure, populate it for BL_HP, BL_HPMAX, BL_ENE, and BL_EXEMAX and switch to it for their percentage calculations. Doesn't seem to break highlighting of 'normal' range values but hasn't been tested for extreme ones.
This commit is contained in:
@@ -1,4 +1,4 @@
|
||||
/* NetHack 3.7 botl.h $NHDT-Date: 1596498528 2020/08/03 23:48:48 $ $NHDT-Branch: NetHack-3.7 $:$NHDT-Revision: 1.34 $ */
|
||||
/* NetHack 3.7 botl.h $NHDT-Date: 1694893330 2023/09/16 19:42:10 $ $NHDT-Branch: NetHack-3.7 $:$NHDT-Revision: 1.37 $ */
|
||||
/* Copyright (c) Michael Allison, 2003 */
|
||||
/* NetHack may be freely redistributed. See license for details. */
|
||||
|
||||
@@ -255,7 +255,7 @@ struct istat_s {
|
||||
boolean percent_matters;
|
||||
short percent_value;
|
||||
unsigned anytype;
|
||||
anything a;
|
||||
anything a, rawval;
|
||||
char *val;
|
||||
int valwidth;
|
||||
enum statusfields idxmax;
|
||||
|
||||
80
src/botl.c
80
src/botl.c
@@ -1,4 +1,4 @@
|
||||
/* NetHack 3.7 botl.c $NHDT-Date: 1685863332 2023/06/04 07:22:12 $ $NHDT-Branch: NetHack-3.7 $:$NHDT-Revision: 1.233 $ */
|
||||
/* NetHack 3.7 botl.c $NHDT-Date: 1694893342 2023/09/16 19:42:22 $ $NHDT-Branch: NetHack-3.7 $:$NHDT-Revision: 1.239 $ */
|
||||
/* Copyright (c) Stichting Mathematisch Centrum, Amsterdam, 1985. */
|
||||
/*-Copyright (c) Michael Allison, 2006. */
|
||||
/* NetHack may be freely redistributed. See license for details. */
|
||||
@@ -529,12 +529,12 @@ static void status_hilites_viewall(void);
|
||||
|
||||
#define INIT_BLSTAT(name, fmtstr, anytyp, wid, fld) \
|
||||
{ name, fmtstr, 0L, FALSE, FALSE, 0, anytyp, \
|
||||
{ (genericptr_t) 0 }, (char *) 0, \
|
||||
wid, -1, fld INIT_THRESH }
|
||||
{ (genericptr_t) 0 }, { (genericptr_t) 0 }, (char *) 0, \
|
||||
wid, -1, fld INIT_THRESH }
|
||||
#define INIT_BLSTATP(name, fmtstr, anytyp, wid, maxfld, fld) \
|
||||
{ name, fmtstr, 0L, FALSE, TRUE, 0, anytyp, \
|
||||
{ (genericptr_t) 0 }, (char *) 0, \
|
||||
wid, maxfld, fld INIT_THRESH }
|
||||
{ (genericptr_t) 0 }, { (genericptr_t) 0 }, (char *) 0, \
|
||||
wid, maxfld, fld INIT_THRESH }
|
||||
|
||||
/* If entries are added to this, botl.h will require updating too.
|
||||
'max' value of BL_EXP gets special handling since the percentage
|
||||
@@ -795,10 +795,12 @@ bot_via_windowport(void)
|
||||
|
||||
/* Hit points */
|
||||
i = Upolyd ? u.mh : u.uhp;
|
||||
if (i < 0)
|
||||
if (i < 0) /* gameover sets u.uhp to -1 */
|
||||
i = 0;
|
||||
gb.blstats[idx][BL_HP].rawval.a_int = i;
|
||||
gb.blstats[idx][BL_HP].a.a_int = min(i, 9999);
|
||||
i = Upolyd ? u.mhmax : u.uhpmax;
|
||||
gb.blstats[idx][BL_HPMAX].rawval.a_int = i;
|
||||
gb.blstats[idx][BL_HPMAX].a.a_int = min(i, 9999);
|
||||
|
||||
/* Dungeon level. */
|
||||
@@ -831,7 +833,9 @@ bot_via_windowport(void)
|
||||
gv.valset[BL_GOLD] = TRUE; /* indicate val already set */
|
||||
|
||||
/* Power (magical energy) */
|
||||
gb.blstats[idx][BL_ENE].rawval.a_int = u.uen;
|
||||
gb.blstats[idx][BL_ENE].a.a_int = min(u.uen, 9999);
|
||||
gb.blstats[idx][BL_ENEMAX].rawval.a_int = u.uenmax;
|
||||
gb.blstats[idx][BL_ENEMAX].a.a_int = min(u.uenmax, 9999);
|
||||
|
||||
/* Armor class */
|
||||
@@ -1241,9 +1245,9 @@ eval_notify_windowport_field(
|
||||
|| (fld == BL_HP && iflags.wc2_hitpointbar)) {
|
||||
fldmax = curr->idxmax;
|
||||
pc = (fldmax == BL_EXP) ? exp_percentage()
|
||||
: (fldmax >= 0 && fldmax < MAXBLSTATS)
|
||||
? percentage(curr, &gb.blstats[idx][fldmax])
|
||||
: 0; /* bullet proofing; can't get here */
|
||||
: (fldmax >= 0 && fldmax < MAXBLSTATS)
|
||||
? percentage(curr, &gb.blstats[idx][fldmax])
|
||||
: 0; /* bullet proofing; can't get here */
|
||||
if (pc != prev->percent_value)
|
||||
chg = (pc < prev->percent_value) ? -1 : 1;
|
||||
curr->percent_value = pc;
|
||||
@@ -1504,7 +1508,9 @@ init_blstats(void)
|
||||
static int
|
||||
compare_blstats(struct istat_s *bl1, struct istat_s *bl2)
|
||||
{
|
||||
int anytype, result = 0;
|
||||
anything *a1, *a2;
|
||||
boolean use_rawval;
|
||||
int anytype, fld, result = 0;
|
||||
|
||||
if (!bl1 || !bl2) {
|
||||
panic("compare_blstat: bad istat pointer %s, %s",
|
||||
@@ -1520,44 +1526,50 @@ compare_blstats(struct istat_s *bl1, struct istat_s *bl2)
|
||||
fmt_ptr((genericptr_t) bl2->a.a_void));
|
||||
}
|
||||
|
||||
fld = bl1->fld;
|
||||
use_rawval = (fld == BL_HP || fld == BL_HPMAX
|
||||
|| fld == BL_ENE || fld == BL_ENEMAX);
|
||||
a1 = use_rawval ? &bl1->rawval : &bl1->a;
|
||||
a2 = use_rawval ? &bl2->rawval : &bl2->a;
|
||||
|
||||
switch (anytype) {
|
||||
case ANY_INT:
|
||||
result = (bl1->a.a_int < bl2->a.a_int) ? 1
|
||||
: (bl1->a.a_int > bl2->a.a_int) ? -1 : 0;
|
||||
result = (a1->a_int < a2->a_int) ? 1
|
||||
: (a1->a_int > a2->a_int) ? -1 : 0;
|
||||
break;
|
||||
case ANY_IPTR:
|
||||
result = (*bl1->a.a_iptr < *bl2->a.a_iptr) ? 1
|
||||
: (*bl1->a.a_iptr > *bl2->a.a_iptr) ? -1 : 0;
|
||||
result = (*a1->a_iptr < *a2->a_iptr) ? 1
|
||||
: (*a1->a_iptr > *a2->a_iptr) ? -1 : 0;
|
||||
break;
|
||||
case ANY_LONG:
|
||||
result = (bl1->a.a_long < bl2->a.a_long) ? 1
|
||||
: (bl1->a.a_long > bl2->a.a_long) ? -1 : 0;
|
||||
result = (a1->a_long < a2->a_long) ? 1
|
||||
: (a1->a_long > a2->a_long) ? -1 : 0;
|
||||
break;
|
||||
case ANY_LPTR:
|
||||
result = (*bl1->a.a_lptr < *bl2->a.a_lptr) ? 1
|
||||
: (*bl1->a.a_lptr > *bl2->a.a_lptr) ? -1 : 0;
|
||||
result = (*a1->a_lptr < *a2->a_lptr) ? 1
|
||||
: (*a1->a_lptr > *a2->a_lptr) ? -1 : 0;
|
||||
break;
|
||||
case ANY_UINT:
|
||||
result = (bl1->a.a_uint < bl2->a.a_uint) ? 1
|
||||
: (bl1->a.a_uint > bl2->a.a_uint) ? -1 : 0;
|
||||
result = (a1->a_uint < a2->a_uint) ? 1
|
||||
: (a1->a_uint > a2->a_uint) ? -1 : 0;
|
||||
break;
|
||||
case ANY_UPTR:
|
||||
result = (*bl1->a.a_uptr < *bl2->a.a_uptr) ? 1
|
||||
: (*bl1->a.a_uptr > *bl2->a.a_uptr) ? -1 : 0;
|
||||
result = (*a1->a_uptr < *a2->a_uptr) ? 1
|
||||
: (*a1->a_uptr > *a2->a_uptr) ? -1 : 0;
|
||||
break;
|
||||
case ANY_ULONG:
|
||||
result = (bl1->a.a_ulong < bl2->a.a_ulong) ? 1
|
||||
: (bl1->a.a_ulong > bl2->a.a_ulong) ? -1 : 0;
|
||||
result = (a1->a_ulong < a2->a_ulong) ? 1
|
||||
: (a1->a_ulong > a2->a_ulong) ? -1 : 0;
|
||||
break;
|
||||
case ANY_ULPTR:
|
||||
result = (*bl1->a.a_ulptr < *bl2->a.a_ulptr) ? 1
|
||||
: (*bl1->a.a_ulptr > *bl2->a.a_ulptr) ? -1 : 0;
|
||||
result = (*a1->a_ulptr < *a2->a_ulptr) ? 1
|
||||
: (*a1->a_ulptr > *a2->a_ulptr) ? -1 : 0;
|
||||
break;
|
||||
case ANY_STR:
|
||||
result = sgn(strcmp(bl1->val, bl2->val));
|
||||
break;
|
||||
case ANY_MASK32:
|
||||
result = (bl1->a.a_ulong != bl2->a.a_ulong);
|
||||
result = (a1->a_ulong != a2->a_ulong);
|
||||
break;
|
||||
default:
|
||||
result = 1;
|
||||
@@ -1655,15 +1667,18 @@ s_to_anything(anything *a, char *buf, int anytype)
|
||||
}
|
||||
#endif /* STATUS_HILITES */
|
||||
|
||||
/* integer percentage is 100 * bl->a / maxbl->a */
|
||||
static int
|
||||
percentage(struct istat_s *bl, struct istat_s *maxbl)
|
||||
{
|
||||
int result = 0;
|
||||
int anytype;
|
||||
int ival;
|
||||
int ival, mval;
|
||||
long lval;
|
||||
unsigned uval;
|
||||
unsigned long ulval;
|
||||
int fld = bl->fld;
|
||||
boolean use_rawval = (fld == BL_HP || fld == BL_ENE);
|
||||
|
||||
if (!bl || !maxbl) {
|
||||
impossible("percentage: bad istat pointer %s, %s",
|
||||
@@ -1676,8 +1691,12 @@ percentage(struct istat_s *bl, struct istat_s *maxbl)
|
||||
if (maxbl->a.a_void) {
|
||||
switch (anytype) {
|
||||
case ANY_INT:
|
||||
ival = bl->a.a_int;
|
||||
result = ((100 * ival) / maxbl->a.a_int);
|
||||
/* HP and energy are int so this is the only case that cares
|
||||
about 'rawval'; for them, we use that rather than their
|
||||
potentially truncated (to 9999) display value */
|
||||
ival = use_rawval ? bl->rawval.a_int : bl->a.a_int;
|
||||
mval = use_rawval ? maxbl->rawval.a_int : maxbl->a.a_int;
|
||||
result = ((100 * ival) / mval);
|
||||
break;
|
||||
case ANY_LONG:
|
||||
lval = bl->a.a_long;
|
||||
@@ -1749,6 +1768,7 @@ exp_percentage(void)
|
||||
curval.a = maxval.a = cg.zeroany;
|
||||
curval.a.a_long = exp_val;
|
||||
maxval.a.a_long = nxt_exp_val;
|
||||
curval.fld = maxval.fld = BL_EXP; /* (neither BL_HP nor BL_ENE) */
|
||||
/* maximum delta between levels is 10000000; calculation of
|
||||
100 * (10000000 - N) / 10000000 fits within 32-bit long */
|
||||
res = percentage(&curval, &maxval);
|
||||
|
||||
Reference in New Issue
Block a user